travetto: Dependency Injection
Install: primary
$ npm install @travetto/di
Dependency injection
is a framework primitive. When used in conjunction with automatic file scanning, it provides for handling of application dependency wiring. Due to the nature of typescript
and type erasure of interfaces, dependency injection only supports class
es as type signafiers. The primary goal of dependency injection is to allow for separation of concerns of object creation and it's usage.
Declaration
The @Injectable
and @InjectableFactory
decorators provide the registration of dependencies. Dependency declaration revolves around exposing class
es and subtypes thereof to provide necessary functionality. Additionally, the framework will utilize dependencies to satisfy contracts with various backends (e.g. ModelMongoSource
provides itself as an injectable candidate for ModelSource
).
Code: Example @Injectable
@Injectable()
class CustomService {
async coolOperation() {
... work ...
}
}
When declaring a dependency, you can also provide a token to allow for multiple instances of the dependency to be defined. This can be used in many situations:
Code: Example @Injectable with multiple targets
@Injectable()
class CustomService {
async coolOperation() {
... work ...
}
}
const CUSTOM2 = Symbol('custom2');
@Injectable({ target: CustomService, symbol: CUSTOM2 })
class CustomService2 extends CustomService {
async coolOperation() {
await super.coolOperation();
}
}
As you can see, the target
field is also set, which indicates to the dependency registration process what class
the injectable is compatible with. Additionally, when using abstract
classes, the parent class
is always considered as a valid candidate type.
Code: Example @Injectable with target via abstract class
abstract class BaseService {
abstract work():Promise<void>;
}
@Injectable()
class SpecificService extends BaseService {
async work() {
}
}
In this scenario, SpecificService
is a valid candidate for BaseService
due to the abstract inheritance. Sometimes, you may want to provide a slight variation to a dependency without extending a class. To this end, the @InjectableFactory
decorator denotes a static
class method that produces an @Injectable
.
Code: Example @InjectableFactory, return type defines target class
class Config {
@InjectableFactory()
static initService(): CoolService {
return new CoolService();
}
}
Given the static
method initService
, the function will be provided as a valid candidate for CoolService
. Instead of calling the constructor of the type directly, this function will work as a factory for producing the injectable.
NOTE Due to the lack of typechecker in the Compiler
for performance reasons, the return type on the factory method is mandatory. Without it, the code will not know what the expected target type should be.
NOTE Other modules are able to provide aliases to @Injectable
that also provide additional functionality. For example, the @Config
or the @Controller
decorator registers the associated class as an injectable element.
Injection
Once all of your necessary dependencies are defined, now is the time to provide those @Injectable
instances to your code. There are three primary methods for injection:
The @Inject
decorator, which denotes a desire to inject a value directly. These will be set post construction.
Code: Example @Injectable with dependencies as @Inject fields
@Injectable()
class CustomService {
@Inject()
private dependentService: DependentService;
async coolOperation() {
await this.dependentService.doWork();
}
}
The @Injectable
constructor params, which will be provided as the instance is being constructed.
Code: Example @Injectable with dependencies in constructor
@Injectable()
class CustomService {
constructor (private dependentService: DependentService) {}
async coolOperation() {
await this.dependentService.doWork();
}
}
Via @InjectableFactory
params, which are comparable to constructor params
Code: Example @InjectableFactory with parameters as dependencies
class Config {
@InjectableFactory()
static initService(dependentService: DependentService): CustomService {
return new CustomService(dependentService);
}
}
Manual Invocation
Some times you will need to lookup a dependency dynamically, or you want to control the injection process at a more granular level. To achieve that you will need to directly access the DependencyRegistry
. The registry allows for requesting a dependency by class reference:
Code: Example of manual lookup
@Injectable()
class Complex {}
class ManualLookup {
async invoke() {
const complex = await DependencyRegistry.getInstance(Complex);
}
}
@Application
Given that dependency injection is generally a pre-requisite for application execution, it stands as the primary entrypoint for application invocation. The Base
provides a simplistic bootstrap to allow for the application to run, but that is not sufficient for more complex applications.
The module provides a decorator, @Application
who's job is to register entry points into the application. For example:
Code: Example of @Application target
import { Application, Inject, Injectable } from '../';
@Injectable()
class Server {
name = 'roger';
async launch() {
...
}
}
@Application('simple')
class SimpleApp {
@Inject()
server: Server
async run() {
return this.server.launch();
}
}
The @Application
decorator exposes some additional functionality, which can be used to launch the application.
.run()
Arguments
The arguments specified in the run
method, will now be able to be specified when invoking the application from the command line. For instance:
Code: Simple Entrypoint with Parameters
@Application('simple')
class SimpleApp {
async run(domain: string, port = 3000) {
console.log('Launching', domain, 'on port', port);
}
}
These command line invocation of travetto run
would look like:
Terminal: Sample CLI Output
$ travetto run
....
● [e2e] simple
----------------------------------
usage: simple [domain] (port=3000)
To invoke the simple
application, you need to pass domain
where port is optional with a default.
Terminal: Invoke Simple
$ travetto run simple mydomain.biz 4000
[INFO] Launching mydomain.biz on port 4000
Additionally, the parameters will be type checked, to ensure proper evaluation.
Terminal: Invoke Simple with bad port
$ travetto run simple mydomain.biz orange
usage: simple domain (string), port=[3000] (number)
The types are inferred from the .run()
method parameters, but can be overridden in the @Application
annotation to support customization. Only primitive types are supported:
number
- Float or decimalstring
- Default if no type is specifiedboolean
- true(yes/on/1) and false(no/off/0)union
- Type unions of the same type (string_a|string_b
or 1|2|3|4
)
Customizing the types is done by name, and allows for greater control:
Code: Complex Entrypoint with Customization
@Application('complex', {
watchable: true,
paramMap: {
domain: {
title: 'Domain Name',
type: 'string',
subtype: 'url'
},
port : {
title: 'Server Port',
def: 3000
}
}
})
class ComplexApp {
async run(domain: string, port: number) {
console.log('Launching', domain, 'on port', port);
}
}